Skip to content

Method: indexedPairStream(Iterable, IntFunction)

1: /*
2: * *************************************************************************************************************************************************************
3: *
4: * TheseFoolishThings: Miscellaneous utilities
5: * http://tidalwave.it/projects/thesefoolishthings
6: *
7: * Copyright (C) 2009 - 2025 by Tidalwave s.a.s. (http://tidalwave.it)
8: *
9: * *************************************************************************************************************************************************************
10: *
11: * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
12: * You may obtain a copy of the License at
13: *
14: * http://www.apache.org/licenses/LICENSE-2.0
15: *
16: * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
17: * CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
18: *
19: * *************************************************************************************************************************************************************
20: *
21: * git clone https://bitbucket.org/tidalwave/thesefoolishthings-src
22: * git clone https://github.com/tidalwave-it/thesefoolishthings-src
23: *
24: * *************************************************************************************************************************************************************
25: */
26: package it.tidalwave.util;
27:
28: import javax.annotation.Nonnegative;
29: import javax.annotation.Nonnull;
30: import javax.annotation.concurrent.Immutable;
31: import javax.annotation.concurrent.NotThreadSafe;
32: import java.util.Map;
33: import java.util.concurrent.atomic.AtomicInteger;
34: import java.util.function.IntFunction;
35: import java.util.function.IntUnaryOperator;
36: import java.util.stream.Collector;
37: import java.util.stream.Collectors;
38: import java.util.stream.IntStream;
39: import java.util.stream.Stream;
40: import java.util.stream.StreamSupport;
41: import lombok.EqualsAndHashCode;
42: import lombok.Getter;
43: import lombok.RequiredArgsConstructor;
44: import lombok.ToString;
45:
46: /***************************************************************************************************************************************************************
47: *
48: * A value object that contains a pair of values. Some factory methods allow creating pairs out of existing collections
49: * or arrays associating an index.
50: *
51: * @author Fabrizio Giudici
52: * @since 3.2-ALPHA-6
53: * @it.tidalwave.javadoc.draft
54: *
55: **************************************************************************************************************************************************************/
56: @Getter @Immutable @RequiredArgsConstructor(staticName = "of") @ToString @EqualsAndHashCode
57: public class Pair<A, B>
58: {
59: /** A base 0 index rebaser. */
60: public static final IntUnaryOperator BASE_0 = i -> i;
61:
62: /** A base 1 index rebaser. */
63: public static final IntUnaryOperator BASE_1 = i -> i + 1;
64:
65: @Nonnull
66: public final A a;
67:
68: @Nonnull
69: public final B b;
70:
71: /***********************************************************************************************************************************************************
72: * Creates a {@link Stream} of {@code Pair}s composed of a given fixed value and another element taken from another
73: * {@link Stream}.
74: *
75: * @param <T> the type of the value
76: * @param <U> the type of the {@code Stream}
77: * @param value the value
78: * @param stream the {@code Stream}
79: * @return the {@code Stream} of {@code Pair}s
80: * @since 3.2-ALPHA-12
81: **********************************************************************************************************************************************************/
82: @Nonnull
83: public static <T, U> Stream<Pair<T, U>> pairStream (@Nonnull final T value,
84: @Nonnull final Stream<? extends U> stream)
85: {
86: return stream.map(object -> Pair.of(value, object));
87: }
88:
89: /***********************************************************************************************************************************************************
90: * Creates a {@link Stream} of {@code Pair}s composed of a given fixed value and an integer in the given range.
91: *
92: * @param <T> the type of the value
93: * @param value the value
94: * @param from the first value of the integer {@code Stream} (included)
95: * @param to the last value of the integer {@code Stream} (excluded)
96: * @return the {@code Stream} of {@code Pair}s
97: * @since 3.2-ALPHA-12
98: **********************************************************************************************************************************************************/
99: @Nonnull
100: public static <T> Stream<Pair<T, Integer>> pairRange (@Nonnull final T value,
101: @Nonnegative final int from,
102: @Nonnegative final int to)
103: {
104: return pairStream(value, IntStream.range(from, to).boxed());
105: }
106:
107: /***********************************************************************************************************************************************************
108: * Creates a {@link Stream} of {@code Pair}s composed of a given fixed value and an integer in the given range.
109: *
110: * @param <T> the type of the value
111: * @param value the value
112: * @param from the first value of the integer {@code Stream} (included)
113: * @param to the last value of the integer {@code Stream} (included)
114: * @return the {@code Stream} of {@code Pair}s
115: * @since 3.2-ALPHA-12
116: **********************************************************************************************************************************************************/
117: @Nonnull
118: public static <T> Stream<Pair<T, Integer>> pairRangeClosed (@Nonnull final T value,
119: @Nonnegative final int from,
120: @Nonnegative final int to)
121: {
122: return pairStream(value, IntStream.rangeClosed(from, to).boxed());
123: }
124:
125: /***********************************************************************************************************************************************************
126: * Returns a {@link Stream} out of the elements in a given array made of {@link Pair}s {@code (index, value)}.
127: *
128: * @param <T> the type of the elements
129: * @param array the array
130: * @return the stream
131: **********************************************************************************************************************************************************/
132: @Nonnull
133: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final T[] array)
134: {
135: return indexedPairStream(array, BASE_0);
136: }
137:
138: /***********************************************************************************************************************************************************
139: * Returns a {@link Stream} out of the elements in the array, made of {@link Pair}s {@code (index, value)}. The
140: * index can be rebased.
141: *
142: * @param <T> the type of the elements
143: * @param array the array
144: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
145: * @return the stream
146: **********************************************************************************************************************************************************/
147: @Nonnull
148: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final T[] array,
149: @Nonnull final IntUnaryOperator rebaser)
150: {
151: return indexedPairStream(array, rebaser, i -> i);
152: }
153:
154: /***********************************************************************************************************************************************************
155: * Returns a {@link Stream} out of the elements in a given array made of {@link Pair}s {@code (index, value)}. The
156: * index is transformed with the given function.
157: *
158: * @param <I> the type of the transformed index
159: * @param <T> the type of the elements
160: * @param array the array
161: * @param indexTransformer the transformer of the index
162: * @return the stream
163: **********************************************************************************************************************************************************/
164: @Nonnull
165: public static <I, T> Stream<Pair<I, T>> indexedPairStream (@Nonnull final T[] array,
166: @Nonnull final IntFunction<? extends I> indexTransformer)
167: {
168: return indexedPairStream(array, BASE_0, indexTransformer);
169: }
170:
171: /***********************************************************************************************************************************************************
172: * Returns a {@link Stream} out of the elements in the array, made of {@link Pair}s {@code (index, value)}. The
173: * index can be rebased and transformed with specific functions.
174: *
175: * @param <T> the type of the elements
176: * @param <I> the type of the transformed index
177: * @param array the array
178: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
179: * @param indexTransformer the transformer of the index
180: * @return the stream
181: **********************************************************************************************************************************************************/
182: @Nonnull
183: public static <T, I> Stream<Pair<I, T>> indexedPairStream (@Nonnull final T[] array,
184: @Nonnull final IntUnaryOperator rebaser,
185: @Nonnull final IntFunction<? extends I> indexTransformer)
186: {
187: return IntStream.range(0, array.length).mapToObj(i -> of(indexTransformer.apply(rebaser.applyAsInt(i)), array[i]));
188: }
189:
190: /***********************************************************************************************************************************************************
191: * Returns a {@link Stream} out of the elements in a given {@link Iterable} made of {@link Pair}s {@code (index,
192: * value)}.
193: *
194: * @param <T> the type of the elements
195: * @param iterable the iterable
196: * @return the stream
197: **********************************************************************************************************************************************************/
198: @Nonnull
199: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final Iterable<? extends T> iterable)
200: {
201: return indexedPairStream(iterable, BASE_0);
202: }
203:
204: /***********************************************************************************************************************************************************
205: * Returns a {@link Stream} out of the elements in a given {@link Iterable} made of {@link Pair}s {@code (index,
206: * value)}. The index can be rebased.
207: *
208: * @param <T> the type of the elements
209: * @param iterable the iterable
210: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
211: * @return the stream
212: **********************************************************************************************************************************************************/
213: @Nonnull
214: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final Iterable<? extends T> iterable,
215: @Nonnull final IntUnaryOperator rebaser)
216: {
217: return indexedPairStream(iterable, rebaser, i -> i);
218: }
219:
220: /***********************************************************************************************************************************************************
221: * Returns a {@link Stream} out of the elements in a given {@link Iterable} made of {@link Pair}s {@code (index,
222: * value)}. The index is transformed with the given function.
223: *
224: * @param <I> the type of the transformed index
225: * @param <T> the type of the elements
226: * @param iterable the iterable
227: * @param indexTransformer the transformer of the index
228: * @return the stream
229: **********************************************************************************************************************************************************/
230: @Nonnull
231: public static <I, T> Stream<Pair<I, T>> indexedPairStream (@Nonnull final Iterable<? extends T> iterable,
232: @Nonnull final IntFunction<? extends I> indexTransformer)
233: {
234: return indexedPairStream(iterable, BASE_0, indexTransformer);
235: }
236:
237: /***********************************************************************************************************************************************************
238: * Returns a {@link Stream} out of the elements returned by an iterable, made of {@link Pair}s
239: * {@code (index, value)}. The index is rebased and transformed with specific functions.
240: *
241: * @param <T> the type of the elements
242: * @param <I> the type of the transformed index
243: * @param iterable the iterable
244: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
245: * @param indexTransformer the transformer of the index
246: * @return the stream
247: **********************************************************************************************************************************************************/
248: @Nonnull
249: public static <I, T> Stream<Pair<I, T>> indexedPairStream (@Nonnull final Iterable<? extends T> iterable,
250: @Nonnull final IntUnaryOperator rebaser,
251: @Nonnull final IntFunction<? extends I> indexTransformer)
252: {
253: return new Factory<I, T>().stream(iterable, rebaser, indexTransformer);
254: }
255:
256: /***********************************************************************************************************************************************************
257: * Returns a {@link Stream} out of the elements in a given {@link Stream} made of {@link Pair}s {@code (index,
258: * value)}.
259: *
260: * @param <T> the type of the elements
261: * @param stream the stream
262: * @return the stream
263: * @since 3.2-ALPHA-12
264: **********************************************************************************************************************************************************/
265: @Nonnull @SuppressWarnings("unchecked")
266: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final Stream<? extends T> stream)
267: {
268: return indexedPairStream(((Stream<T>)stream)::iterator);
269: }
270:
271: /***********************************************************************************************************************************************************
272: * Returns a {@link Stream} out of the elements in a given {@link Stream} made of {@link Pair}s {@code (index,
273: * value)}. The index can be rebased.
274: *
275: * @param <T> the type of the elements
276: * @param stream the stream
277: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
278: * @return the stream
279: * @since 3.2-ALPHA-12
280: **********************************************************************************************************************************************************/
281: @Nonnull @SuppressWarnings("unchecked")
282: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnull final Stream<? extends T> stream,
283: @Nonnull final IntUnaryOperator rebaser)
284: {
285: return indexedPairStream(((Stream<T>)stream)::iterator, rebaser);
286: }
287:
288: /***********************************************************************************************************************************************************
289: * Returns a {@link Stream} out of the elements in a given {@link Stream} made of {@link Pair}s {@code (index,
290: * value)}. The index is transformed with the given function.
291: *
292: * @param <I> the type of the transformed index
293: * @param <T> the type of the elements
294: * @param stream the stream
295: * @param indexTransformer the transformer of the index
296: * @return the stream
297: * @since 3.2-ALPHA-12
298: **********************************************************************************************************************************************************/
299: @SuppressWarnings("unchecked")
300: @Nonnull
301: public static <I, T> Stream<Pair<I, T>> indexedPairStream (@Nonnull final Stream<? extends T> stream,
302: @Nonnull final IntFunction<? extends I> indexTransformer)
303: {
304: return indexedPairStream(((Stream<T>)stream)::iterator, indexTransformer);
305: }
306:
307: /***********************************************************************************************************************************************************
308: * Returns a {@link Stream} out of the elements returned by a Stream, made of {@link Pair}s
309: * {@code (index, value)}. The index is rebased and transformed with specific functions.
310: *
311: * @param <T> the type of the elements
312: * @param <I> the type of the transformed index
313: * @param stream the stream
314: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
315: * @param indexTransformer the transformer of the index
316: * @return the stream
317: * @since 3.2-ALPHA-12
318: **********************************************************************************************************************************************************/
319: @SuppressWarnings("unchecked")
320: @Nonnull
321: public static <I, T> Stream<Pair<I, T>> indexedPairStream (@Nonnull final Stream<? extends T> stream,
322: @Nonnull final IntUnaryOperator rebaser,
323: @Nonnull final IntFunction<? extends I> indexTransformer)
324: {
325: return indexedPairStream(((Stream<T>)stream)::iterator, rebaser, indexTransformer);
326: }
327:
328: /***********************************************************************************************************************************************************
329: * Returns a {@link Stream} out of the elements returned by a supplier, made of {@link Pair}s
330: * {@code (index, value)}.
331: *
332: * @param <T> the type of the elements
333: * @param from the first index (included)
334: * @param to the last index (excluded)
335: * @param valueSupplier the supplier of values
336: * @return the stream
337: **********************************************************************************************************************************************************/
338: @Nonnull
339: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnegative final int from,
340: @Nonnegative final int to,
341: @Nonnull final IntFunction<? extends T> valueSupplier)
342: {
343: return indexedPairStream(from, to, valueSupplier, BASE_0, i -> i);
344: }
345:
346: /***********************************************************************************************************************************************************
347: * Returns a {@link Stream} out of the elements returned by a supplier, made of {@link Pair}s
348: * {@code (index, value)}.
349: *
350: * @param <T> the type of the elements
351: * @param from the first index (included)
352: * @param to the last index (excluded)
353: * @param valueSupplier the supplier of values
354: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
355: * @return the stream
356: **********************************************************************************************************************************************************/
357: @Nonnull
358: public static <T> Stream<Pair<Integer, T>> indexedPairStream (@Nonnegative final int from,
359: @Nonnegative final int to,
360: @Nonnull final IntFunction<? extends T> valueSupplier,
361: @Nonnull final IntUnaryOperator rebaser)
362: {
363: return indexedPairStream(from, to, valueSupplier, rebaser, i -> i);
364: }
365:
366: /***********************************************************************************************************************************************************
367: * Returns a {@link Stream} out of the elements returned by a supplier, made of {@link Pair}s
368: * {@code (index, value)}. The index can be rebased and transformed with specific functions.
369: *
370: * @param <I> the type of the transformed index
371: * @param <T> the type of the elements
372: * @param from the first index (included)
373: * @param to the last index (excluded)
374: * @param valueSupplier the supplier of values
375: * @param rebaser the rebaser of the index (BASE_0, BASE_1 or a similar function)
376: * @param indexTransformer the transformer of the index
377: * @return the stream
378: **********************************************************************************************************************************************************/
379: @Nonnull
380: public static <T, I> Stream<Pair<I, T>> indexedPairStream (@Nonnegative final int from,
381: @Nonnegative final int to,
382: @Nonnull final IntFunction<? extends T> valueSupplier,
383: @Nonnull final IntUnaryOperator rebaser,
384: @Nonnull final IntFunction<? extends I> indexTransformer)
385: {
386: return IntStream.range(from, to).mapToObj(i -> Pair.of(indexTransformer.apply(rebaser.applyAsInt(i)),
387: valueSupplier.apply(i)));
388: }
389:
390: /***********************************************************************************************************************************************************
391: * A {@link Collector} that produces a {@link Map} whose key is field {@code a} and value field {@code b}. Use
392: * with {@link Stream#collect(Collector)}.
393: *
394: * @param <A> the type of the former element of the pair
395: * @param <B> the type of the latter element of the pair
396: * @return the {@code Collector}
397: **********************************************************************************************************************************************************/
398: @Nonnull
399: public static <A, B> Collector<Pair<A, B>, ?, Map<A, B>> pairsToMap()
400: {
401: return Collectors.toMap(p -> p.a, p -> p.b);
402: }
403:
404: /***********************************************************************************************************************************************************
405: * Zips two streams into a stream of {@link Pair}s.
406: *
407: * @param streamA the first {@link Stream}
408: * @param streamB the second {@link Stream}
409: * @param <A> the type of elements of the first {@link Stream}
410: * @param <B> the type of elements of the second {@link Stream}
411: * @return the zipped {@link Stream}
412: * @since 3.2-ALPHA-17 (since 3.2-ALPHA-12 was in {@code StreamOperations}
413: **********************************************************************************************************************************************************/
414: @Nonnull
415: public static <A, B> Stream<Pair<A, B>> zip (@Nonnull final Stream<? extends A> streamA,
416: @Nonnull final Stream<? extends B> streamB)
417: {
418: return StreamUtils.zip(streamA, streamB);
419: }
420:
421: @NotThreadSafe
422: static final class Factory<I, T>
423: {
424: private final AtomicInteger n = new AtomicInteger(0);
425:
426: @Nonnull
427: public Stream<Pair<I, T>> stream (@Nonnull final Iterable<? extends T> iterable,
428: @Nonnull final IntUnaryOperator rebaser,
429: @Nonnull final IntFunction<? extends I> indexFunction)
430: {
431: return StreamSupport.stream(iterable.spliterator(), false)
432: .map(o -> Pair.of(indexFunction.apply(rebaser.applyAsInt(n.getAndIncrement())), o));
433: }
434: }
435: }